Frontend Forever App
We have a mobile app for you to download and use. And you can unlock many features in the app.
Get it now
Intall Later
Run
HTML
CSS
Javascript
Output
@import url("https://fonts.googleapis.com/css2?family=Pixelify+Sans:wght@700&display=swap"); html, body { margin: 0; padding: 0; } canvas { display: block; cursor: crosshair; overflow: none; } .footer { position: fixed; bottom: 10px; right: 10px; color: white; font-size: 12px; text-align: right; font-family: monospace; pointer-events: none; } .footer a { color: white; text-decoration: none; }
const tileSize = 32; let spiders = []; let player; let spiderSpawnTime = 300; let spiderMaxSpeed = 1.2; let frame = 0; let score = 0; // Preload images let playerSprite; let spiderSprite; function preload() { playerSprite = loadImage( "" ); spiderSprite = loadImage( "" ); } let touches = []; let isMobileDevice = false; function setup() { canvas = createCanvas(windowWidth, windowHeight); canvas.elt.setAttribute("style", "display: block"); pixelDensity(1); player = new Player(); for (let i = 0; i < 42; i++) { spiders.push(new Spider(random(spiderMaxSpeed))); } // Detect if it's a mobile device detectMobile(); if (isMobileDevice) { // Add touch event listeners for player movement and shooting canvas.mousePressed(() => { // Trigger shooting on touch (for mobile devices) if (mouseButton === LEFT) { player.shoot(); } }); // Add touch event listeners for player movement canvas.touchStarted(handleTouch); canvas.touchMoved(handleTouch); canvas.touchEnded(handleTouch); } } function handleTouch(event) { touches = event.touches; } // Detect if it's a mobile device function detectMobile() { if ( /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test( navigator.userAgent ) ) { isMobileDevice = true; } } function windowResized() { resizeCanvas(windowWidth, windowHeight); } function draw() { background(97, 87, 97); rectMode(CENTER); player.draw(); player.update(); player.animate(); for (let i = spiders.length - 1; i >= 0; i--) { spiders[i].draw(); spiders[i].update(); if (spiders[i].ateYou()) { restart(); break; } if (player.hasShot(spiders[i])) { score++; spiders.splice(i, 1); } } if (frame >= random(spiderSpawnTime, spiderSpawnTime * 1.2)) { spiders.push(new Spider(random(spiderMaxSpeed))); spiderSpawnTime *= 0.95; frame = 0; } if (frameCount % 1000 == 0) { spiderMaxSpeed += 0.1; } frame++; // Draw the score text with a pixelated effect textAlign(CENTER); textSize(40); textFont("Pixelify Sans"); // Use a monospace font for a pixel-like appearance fill(255); // White text stroke(0); // Black outline strokeWeight(4); // Adjust the outline thickness text(score, width / 2, 100); if (touches.length === 1) { const touch = touches[0]; let joystickCenterX = width / 6; let joystickCenterY = height - height / 6; let movementVector = createVector( touch.x - joystickCenterX, touch.y - joystickCenterY ); let movementMagnitude = movementVector.mag(); if (movementMagnitude > 20) { movementVector.normalize(); movementVector.mult(4); // Adjust the movement speed player.pos.add(movementVector); } } } function restart() { player = new Player(); spiders = []; spiderSpawnTime = 300; spiderMaxSpeed = 1.2; score = 0; } function mouseClicked() { player.shoot(); } function randomGradient() { let colors = [ color("#b6908d"), color("#eacec4"), color("#af8886"), color("#b09192") ]; shuffle(colors, true); // Randomize the colors return colors[0]; } class Bullet { constructor(x, y, angle, playerSize) { this.pos = createVector(x, y); this.speed = 16; this.angle = angle; this.playerSize = playerSize; this.bulletLength = random(playerSize * 0.4, playerSize * 0.6); this.bulletWidth = random(playerSize * 0.04, playerSize * 0.06); this.circleSize = random(playerSize * 0.06, playerSize * 0.1); this.fillColor = randomGradient(); } draw() { push(); noStroke(); translate(this.pos.x, this.pos.y); rotate(this.angle); // Gradient fill for the bone let fromColor = this.fillColor; let toColor = color("#b09192"); // Use the last color in the gradient as the end color for (let i = 0; i < 3; i++) { let interColor = lerpColor(fromColor, toColor, i / 2); fill(interColor); rectMode(CENTER); rect(0, 0, this.bulletLength, this.bulletWidth, this.circleSize); fromColor = interColor; } // Circles at the ends of the bone fill(fromColor); ellipse(-this.bulletLength / 2, 0, this.circleSize); ellipse(this.bulletLength / 2, 0, this.circleSize); stroke("#5d4352"); // Outline color strokeWeight(2); // Outline thickness pop(); } update() { const cosAngle = cos(this.angle); const sinAngle = sin(this.angle); this.pos.x += this.speed * cosAngle; this.pos.y += this.speed * sinAngle; } } class Player { constructor() { this.pos = createVector(width / 2, height / 2); this.angle = 0; this.bullets = []; this.spriteSheet = playerSprite; this.spriteFrames = 4; this.animationSpeed = 0.15; this.currentFrame = 0; this.isMoving = false; this.animationState = "idle"; this.animationTimer = 0; this.scaleFactor = 1.7; } draw() { push(); translate(this.pos.x, this.pos.y); scale(this.scaleFactor); // Determine animation frame based on the animation state let frameX = this.currentFrame * tileSize; let frameY = 0; // Adjust sprite frame based on the player's facing direction if (this.animationState === "idle" || this.animationState === "moving") { let facingDirection = this.getFacingDirection(); let flipped = false; // Check if the sprite needs flipping if (facingDirection === "left") { facingDirection = "right"; // Use the 'right' frame flipped = true; // Set the flipped flag } if (this.animationState === "idle") { frameY = tileSize * ["down", "right", "up", "left"].indexOf(facingDirection); } else if (this.isMoving) { frameY = tileSize * (3 + ["down", "right", "up", "left"].indexOf(facingDirection)); } // Flip the sprite if necessary if (flipped) { scale(-1, 1); } } // Draw the sprite with the correct rotation imageMode(CENTER); let sprite = this.spriteSheet.get(frameX, frameY, tileSize, tileSize); image(sprite, 0, 0, tileSize, tileSize); pop(); for (let bullet of this.bullets) { bullet.update(); bullet.draw(); } } getFacingDirection() { let angle = atan2(mouseY - this.pos.y, mouseX - this.pos.x); if (angle < -PI / 4 && angle >= (-3 * PI) / 4) { return "up"; // Facing up } else if (angle >= -PI / 4 && angle < PI / 4) { return "right"; // Facing right } else if (angle >= (3 * PI) / 4 || angle < (-3 * PI) / 4) { return "left"; // Facing left } else { return "down"; // Facing down } } animate() { if (this.animationState === "idle" || this.animationState === "moving") { // Increment the animation timer this.animationTimer += this.animationSpeed; // If the timer exceeds the threshold, update the frame if (this.animationTimer >= 1) { this.currentFrame = (this.currentFrame + 1) % this.spriteFrames; this.animationTimer = 0; // Reset the timer } } } update() { let xSpeed = 0; let ySpeed = 0; if (keyIsPressed) { if (keyIsDown(65)) { xSpeed = -2; } if (keyIsDown(68)) { xSpeed = 2; } if (keyIsDown(87)) { ySpeed = -2; } if (keyIsDown(83)) { ySpeed = 2; } } if (xSpeed !== 0 || ySpeed !== 0) { this.isMoving = true; // Set the movement flag to true } else { this.isMoving = false; // Set the movement flag to false } // Update the animation state based on the movement flag if (this.isMoving) { this.animationState = "moving"; } else { this.animationState = "idle"; } this.pos.add(xSpeed, ySpeed); this.angle = atan2(mouseY - this.pos.y, mouseX - this.pos.x); } shoot() { this.bullets.push(new Bullet(this.pos.x, this.pos.y, this.angle, tileSize)); } hasShot(zombie) { for (let i = this.bullets.length - 1; i >= 0; i--) { if ( dist( this.bullets[i].pos.x, this.bullets[i].pos.y, zombie.pos.x, zombie.pos.y ) < 15 ) { this.bullets.splice(i, 1); return true; } } return false; } } class Spider { constructor(speed) { this.speed = speed; this.pos = this.getOffscreenPosition(); this.spriteSheet = spiderSprite; this.spriteFrames = 8; this.currentFrame = 0; this.frameWidth = 64; this.frameHeight = 64; this.animationSpeed = 4; this.frameCounter = 0; this.scaleFactor = random(0.5, 1); } getOffscreenPosition() { let spawnX, spawnY; // Randomly determine whether to spawn from the top, bottom, left, or right let side = floor(random(4)); if (side === 0) { // Spawn from the top spawnX = random(width); spawnY = -10; } else if (side === 1) { // Spawn from the bottom spawnX = random(width); spawnY = height + 10; } else if (side === 2) { // Spawn from the left spawnX = -10; spawnY = random(height); } else { // Spawn from the right spawnX = width + 10; spawnY = random(height); } return createVector(spawnX, spawnY); } draw() { push(); imageMode(CENTER); translate(this.pos.x, this.pos.y); // Increment the frame counter this.frameCounter++; // Check if it's time to change the frame if (this.frameCounter >= this.animationSpeed) { this.currentFrame = (this.currentFrame + 1) % this.spriteFrames; this.frameCounter = 0; // Reset the frame counter } // Scale the drawing with the random scale factor scale(this.scaleFactor); // Draw the appropriate frame let frameX = this.currentFrame * this.frameWidth; image( this.spriteSheet, 0, 0, this.frameWidth, this.frameHeight, frameX, 0, this.frameWidth, this.frameHeight ); pop(); } getAnimationType() { // Determine animation type based on direction if (this.direction === "south") { return 0; } else if (this.direction === "east") { return 1; } else if (this.direction === "north") { return 2; } else if (this.direction === "west") { return 3; } } update() { let difference = p5.Vector.sub(player.pos, this.pos); difference.limit(this.speed); this.pos.add(difference); } ateYou() { return dist(this.pos.x, this.pos.y, player.pos.x, player.pos.y) < 20; } }